package com.huangcy.gulimall.search.thread;

import java.util.concurrent.*;

/**
 * @program: gulimall
 * @description:
 * @author: huangcy
 * @create: 2022-08-24 09:44
 **/
public class ThreadTest {

    public static ExecutorService executor = Executors.newFixedThreadPool(10);

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        System.out.println("main....start..");
//        CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 2;
//            System.out.println("运行结果：" + i);
//        }, executor);

        // 方法成功完成后的感知
//        CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 0;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).whenComplete((result, exception) -> {
//            // 虽然能得到异常信息，但是没法修改返回数据。
//            System.out.println("异步任务成功完成了...结果是：" + result + ";异常是：" + exception);
//        }).exceptionally(throwable -> {
//            // 可以感知异常同时返回默认值
//            return 10;
//        });


        // 方法执行完成后的处理 无论是成功还是失败
//        CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 4;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).handle((res, thr) -> {
//            if (res != null) {
//                return res * 2;
//            }
//            if (thr != null) {
//                return 0;
//            }
//            return 0;
//        });
//        R apply(T t, U u);


        //
        /**.
         *  串行化
         *  1）、thenRun：不能获取到上一步的执行结果，无返回值
         *          thenRunAsync(() -> {
         *                 System.out.println("任务2启动了");
         *             }, executor);
         *  2)、thenAcceptAsync:能接受上一步结果，但是无返回值
         *  3)、thenApplyAsync：能接收上一步的结果，有返回值
         */
//        CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 4;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).thenApplyAsync(res -> {
//            System.out.println("任务2启动了" + res);
//            return "hello" + res;
//        }, executor);
        //    void accept(T t);
        //    R apply(T t);

        // future.get() 阻塞方法

        /**。
         *  两个都完成
         */


//        CompletableFuture<Object> future01 = CompletableFuture.supplyAsync(() -> {
//            System.out.println("任务1线程：" + Thread.currentThread().getId());
//            int i = 10 / 4;
//            System.out.println("任务1结束：");
//            return i;
//        }, executor);
//
//        CompletableFuture<Object> future02 = CompletableFuture.supplyAsync(() -> {
//            System.out.println("任务2线程：" + Thread.currentThread().getId());
//            System.out.println("任务2结束：");
//            try {
//                Thread.sleep(3000);
//            } catch (InterruptedException e) {
//                e.printStackTrace();
//            }
//            return "hello";
//        }, executor);

//       future01.runAfterBothAsync(future02, () -> {
//            System.out.println("任务3开始..");
//        }, executor);

//        future01.thenAcceptBothAsync(future02, (f1, f2) -> {
//            System.out.println("任务3开始.."+f1+"---->"+f2);
//        }, executor);

//        CompletableFuture<String> future = future01.thenCombineAsync(future02, (f1, f2) -> {
//            return f1 + ":" + f2 + "->haha";
//        }, executor);


        /**.
         * 两个任务，只要有一个完成，我们就执行任务3
         * runAfterEitherAsync：不感知结果，自己没有返回值
         * acceptEitherAsync：感知结果，自己没有返回值
         * applyToEitherAsync
         */
//        CompletableFuture<String> future = future01.applyToEitherAsync(future02, res -> {
//            System.out.println("任务3开始" + res);
//            return res.toString() + "->哈哈";
//        }, executor);
//        String s = future.get();

        CompletableFuture<String> futureImg = CompletableFuture.supplyAsync(() -> {
            System.out.println("查询商品的图片信息");
            return "hello.jpg";
        }, executor);

        CompletableFuture<String> futureAttr = CompletableFuture.supplyAsync(() -> {
            System.out.println("查询商品的属性信息");
            return "黑色+256G";
        }, executor);

        CompletableFuture<String> futureDesc = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(3000);
                System.out.println("查询商品的详情信息");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "华为";
        }, executor);

        // 阻塞等待 消耗时间太长
//        futureImg.get();futureAttr.get();futureDesc.get();

//        CompletableFuture<Void> allOf = CompletableFuture.allOf(futureImg, futureAttr, futureDesc);
        CompletableFuture<Object> anyOf = CompletableFuture.anyOf(futureImg, futureAttr, futureDesc);
        anyOf.get(); // 等待所有结果完成


        System.out.println("main....end.." + anyOf.get());


    }

    public void thread(String[] args) throws ExecutionException, InterruptedException {

        System.out.println("thread start");
/**
 *   1)、继承Thread
 *   Thread01 thread = new Thread01();
 *   thread.start();
 */


/**
 *   2)、实现Runnable接口
 *   new Thread(runnable01).start();
 */

/**
 *   3)、实现Callable接口+FutureTask
 *    FutureTask<Integer> futureTask = new FutureTask<>(new Callable01());
 *    new Thread(futureTask).start();
 *    // 阻塞等待整个线程执行完成，获取返回结果
 *    Integer integer = futureTask.get();
 *    System.out.println("thread end" + integer);
 *
 *   4）、线程池[ExecutorService]
 *    给线程池直接提交任务
 *    servic.execute(new Runnable01())
 *      1、创建：
 *          1）、Executors
 *          2)、
 *
 *
 *   区别；
 *       1、2 不能得到返回值。3可以有返回值
 *       1、2、3都不能控制资源
 *       4 可以对线程进行调度和资源控制，性能稳定。
 */

        // 我们以后再业务代码里面，以上三种启动线程都不用。应该将所有的多线程异步任务都交给线程池执行。

        // 当前系统中池只有一两个，每个异步任务，提交给线程池让他自己执行就行
/**
 *    corePoolSize：[5] 核心线程数[一直存在，除非设置(allowCoreThreadTimeOut)]；线程池，创建好以后就准备就绪的线程数量，就等待来接受异步任务去执行。
 *                  5个  Thread thread =new Thread(); thread.start();
 *    maximumPoolSize：[200]最大线程数量;  控制资源并发
 *    keepAliveTime: 存活时间。如果当前正在运行的线程数量 >  core数量。
 *                   释放空闲的线程（maximumPoolSize-corePoolSize）。只要线程空闲大于指定的keepAliveTime
 *    unit：时间单位
 *    BlockingQueue<Runnable> workQueue：阻塞队列。如果任务有很多，就会将目前多的任务放在队列里面。
 *                   只要用线程空闲了，就会去队列里面取出新的任务继续执行。
 *    ThreadFactory threadFactory：线程的创建工厂。
 *    RejectedExecutionHandler handler：拒绝策略，如果队列满了，按照我们指定的拒绝策略，拒绝执行任务。
 *
 *
 *    工作顺序
 *          1）、线程池创建好，准备好core数量核心线程，准备接受任务
 *          1.1、core 满了，就将再进来的任务放入阻塞队列中。空闲的 core 就会自己去阻塞队列获取任务执行
 *          1.2、阻塞队列满了，就直接开新线程执行，最大只能开到 max 指定的数量
 *          1.3、max满了 就用RejectedExecutionHandler拒绝任务
 *          1.4、max都执行完了，有很多空闲，在指定的时间keepAliveTime以后，释放max-core这些线程
 *
 *
 * new LinkedBlockingDeque<>():默认是Integer的最大值。内存不够
 *
 * 一个线程池 core 7； max 20 ，queue：50，100 并发进来怎么分配的；
 * 7个立即执行，50个会进入队列，再开13个进行执行。剩下的30个就使用拒绝策略
 * 如果不想抛弃还要执行CallerRunsPolicy;
 *
 *
 *
 *
 */
        ThreadPoolExecutor executor = new ThreadPoolExecutor(5,
                200,
                10,
                TimeUnit.SECONDS,
                new LinkedBlockingDeque<>(100000),
                Executors.defaultThreadFactory(),
                new ThreadPoolExecutor.AbortPolicy());

//        Executors.newCachedThreadPool() 核心是10，所有都会回收
//        Executors.newFixedThreadPool() 固定大小 core=max 都不可回收
//        Executors.newScheduledThreadPool() 定时任务的线程池
//        Executors.newSingleThreadExecutor() 单线程的线程池，后台从队列里面获取任务，挨个执行
        System.out.println("thread end");

    }


    public static class Thread01 extends Thread {

        @Override
        public void run() {
            System.out.println("当前线程：" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.println("运行结果：" + i);
        }
    }

    public static class Runnable01 implements Runnable {

        @Override
        public void run() {
            System.out.println("当前线程：" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.println("运行结果：" + i);
        }
    }

    public static class Callable01 implements Callable<Integer> {

        @Override
        public Integer call() throws Exception {
            System.out.println("当前线程：" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.println("运行结果：" + i);
            return i;
        }
    }
}
